Skip to content

Conversation

@iluuu1994
Copy link
Member

Previously, this only worked for classes. It worked by preventing early binding, and then declaring the class at runtime. Instead of doing that, we now avoid calling zend_accel_load_script() completely, which prevents the functions from being added to the function table.

This is breaking, and may thus only be applied to master. I will also notify the list and see whether there are concerns.

Fixes GH-16668

Previously, this only worked for classes. It worked by preventing early binding,
and then declaring the class at runtime. Instead of doing that, we now avoid
calling zend_accel_load_script() completely, which prevents the functions from
being added to the function table.

Fixes phpGH-16668
@iluuu1994 iluuu1994 marked this pull request as ready for review November 19, 2024 20:19
@iluuu1994 iluuu1994 requested a review from dstogov as a code owner November 19, 2024 20:19
@dstogov
Copy link
Member

dstogov commented Nov 20, 2024

According to documentation "opcache_compile_file — Compiles and caches a PHP script without executing it". This doesn't say if function and classes are loaded into the current process. Actually, they were loaded and this patch changes the behavior.

The PR doesn't not affect the main desired behavior (opcache priming), but may break some code that relays on the existing behavior (e.g. usage of opcache_compile_file() during preloading). Please, check this (may be I'm wrong).

@iluuu1994
Copy link
Member Author

iluuu1994 commented Nov 20, 2024

You are right. This is what I referred to here:

This is breaking, and may thus only be applied to master. I will also notify the list and see whether there are concerns.

IMO, it doesn't make much sense to register only functions but not classes. This will break when mixing classes and functions, because using the class will trigger the autoloader and then fail due to function redeclaration. Even more confusing is that only rtd class names are registered, so you can't compile the file twice even though the user cannot see the class as being registered. Actually, this is not true, since RTD collisions are silently ignored.

Nonetheless, this should be discussed on the ML because some code might have to adjust.

@danog
Copy link
Contributor

danog commented Nov 21, 2024

I believe the removal of function caching is a bit of an overreaction: it was always known, to those who used it, that preloading will cache functions, and requires an include guard to avoid function redeclaration errors, and in fact it is a very useful feature.

There is nothing wrong with this behaviour, and it does what you expect it to do.

@iluuu1994
Copy link
Member Author

What do you mean by "caching"?

@danog
Copy link
Contributor

danog commented Nov 21, 2024

@iluuu1994 I have confused the linked issue with another issue related to preloads that I encountered and worked around with include guards (amphp/amp#321).

From what I see here, this PR would disable population of the function table when using opcache_compile_file (which is actually correct, I again misinterpeted the initial issue, thinking it was about preloading); this is what I meant by disabling cache (would the function's op_arrays still be stored anywhere to avoid re-parsing though?)

@iluuu1994
Copy link
Member Author

Yes, the functions are still cached in opcache, they just aren't loaded in the current request. That said, I have noticed some inconsistencies with preloading I'll have to examine. Namely, opcache_compile_file() dodges opcache during preloading and just loads it like normal. Skipping function declaration will be a bit more tricky there, we need to be careful not to introduce more inconsistencies.

@iamacarpet
Copy link
Contributor

Great work @iluuu1994 ,

Now that #16551 has been merged for support of opcache.file_cache_read_only, I'm hopeful that when it launches in PHP 8.5, usage of opcache_compile_file() will hopefully become more widespread within Docker workflows.

It's possible to work around the current behaviour, but it isn't ideal:

Ideally, I'd love to see us adopt this change, but otherwise, updating the documentation to reflect the actual behaviour would be useful.

Comment on lines +376 to +381
if (CG(compiler_options) & ZEND_COMPILE_WITHOUT_EXECUTION) {
/* Not a pretty API, but we need to distinguish between successful and
* unsuccessful compilations. */
return (void *) -1;
}

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please, check the situation when from_shared_memory is false.

@dstogov
Copy link
Member

dstogov commented Nov 27, 2024

That said, I have noticed some inconsistencies with preloading I'll have to examine

@iluuu1994 can you please explain that inconsistencies.
I feel that this may break something, but I can't prove this.

Limiting only zend_accel_load_script() may lead to different behavior when persistent_compile_file() fall-backs to original compiler.

@iluuu1994
Copy link
Member Author

can you please explain that inconsistencies.

Yes. During preloading, persistent_compile_file() does not use the normal file compilation path in opcache. Instead, it immediately falls back to accelerator_orig_compile_file. When preloading is enabled, that is preload_compile_file. This will compile the file as normal, and not actually revert the global symbol tables. This could be fixed by mirroring this logic in preload_compile_file().

Limiting only zend_accel_load_script() may lead to different behavior when persistent_compile_file() fall-backs to original compiler.

This is a fair point. What would be the correct behavior in this case? If the script cannot be cached, there's no point in compiling it at all. So probably the function should just abort.

@dstogov
Copy link
Member

dstogov commented Nov 28, 2024

This could be fixed by mirroring this logic in preload_compile_file().

I afraid this may break preloading (linking of classes loaded through opcache_compile_file()).

Limiting only zend_accel_load_script() may lead to different behavior when persistent_compile_file() fall-backs to original compiler.

This is a fair point. What would be the correct behavior in this case? If the script cannot be cached, there's no point in compiling it at all. So probably the function should just abort.

probably yes.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

opcache_compile_file() fails with Cannot redeclare function error when compiling multiple files with the same function

4 participants